# Introduction to Microcontrollers

Timers and Interrupts

# Goals

- Know what a timer is, and what they are used for
- Know what interrupts are, and what they are used for
- Learn how to use interrupts with AVRs
- Learn how to make PWM signals with AVRs

### Timers

- Also referred to as counters
- Main objective of a timer is to count
  - Count up
  - Count down
  - Count up then down
  - You name it
- So what is so special about counting?
  - Any thoughts?



# The Power of Counting

- Periodic interrupts
  - Maybe a piece of code needs to happen periodically
  - Maybe a delay function that is non-blocking via ticks?
- Time-outs
  - Stop a task if it took to long?
- Waveform measurements
  - Frequency
  - Pulse-width
  - Number of events
- Waveform generation
  - Pulse-width
  - Frequency
- Noise cancellation
  - Maybe debounce a button with a single shot counter?
- This is just the possibility of ONE counter
  - By combining multiple we can create very complex circuits in hardware!

#### Luis Arias



# **Definitions**

- CNT
  - Counter value Increments every clk\_per
- TOP
  - Counter maximum value Overflow when CNT = TOP
    - Alternatively BTM for underflow, but TOP is more used
  - Some AVR peripherals call this PER for period
- CMP
  - Compare Toggle/Clear register CNT = CMP
    - Dependent on timer/counter. If Clear is used, register is set on overflow
  - Often used for waveform generation

# Timer as Waveform Generator

- Typically we count UP
  - Reset CNT value on TOP
    - Overflow
- Typically we have one or more CMP registers
  - CMP0, CMP1, ..., CMPn
- Example: PWM
  - Pulse-Width Modulation
  - GPIO is '1' when CNT < CMP</li>
  - GPIO is '0' when CNT > CMP



# Timers as Waveform Generator

- PWM
  - Pulse-Width = ON time
  - Duty-Cycle = ON time / OFF time
- Frequency
  - 1/Period
  - Period = ON time + OFF time
- Timer/counter resolution
  - TOP register value determines
    - Frequency
    - Resolution for PWM
  - Not all duty cycles and frequencies can be synthesized
    - Dithering is an option



# **PWM**

- Control external circuits
- Typical uses
  - LED dimming
  - DACs
  - Motor control
  - Voltage regulation
- Can be combined with software to create complex signals
  - Example: Sinusoidal PWM



# Gotchas

- Some timer counters have double buffering
  - Registers updated on overflow
  - Makes sure timer always works as intended
- Not all timer counters can do everything
  - They are usually designed for some specific tasks
  - This is why we have so many variants



### Timers in AVRs

Many different peripherals

#### Name

- TCA (Most newer AVRs)
- TCB (All newer AVRs)
- TCD (Some newer AVRs)
- TCE (Few newer AVRs)
- TCF (Few newer AVRs)
- RTC (All newer AVRs)
- ++

#### Common use

- Waveform generation
- Waveform measurement / Periodic interrupt
- Motor control
- Waveform generation
- Frequency generation
- Periodic interrupt

### **TCB**

- Simplest timer counter
- AVR128DA48
  - 4 instances!
  - TCB0, TCB1, TCB2, TCB3

We will use it for periodic interrupts today

#### **Peripheral Overview**

The following table shows the peripheral overview of the entire AVR® DA(S) family. Further documentation describes only the AVR128DA28/32/48/64(S) devices.

Table 2. Peripheral Overview

| Feature                            | AVR128DA28(S)<br>AVR64DA28(S)<br>AVR32DA28(S) | AVR128DA32(S)<br>AVR64DA32(S)<br>AVR32DA32(S) | AVR128DA48(5)<br>AVR64DA48(S)<br>AVR32DA48(S) | AVR128DA64(S)<br>AVR64DA64(S) |
|------------------------------------|-----------------------------------------------|-----------------------------------------------|-----------------------------------------------|-------------------------------|
| Pins                               | 28                                            | 32                                            | 48                                            | 64                            |
| Max. Frequency (MHz)               | 24                                            | 24                                            | 24                                            | 24                            |
| 16-bit Timer/Counter type A (TCA)  | 1                                             | 1                                             | 2                                             | 2                             |
| 16-bit Timer/Counter type B (TCB)  | 3                                             | 3                                             | 4                                             | 5                             |
| 12-bit Timer/Counter type D (TCD)  | 1                                             | 1                                             | 1                                             | 1                             |
| Real-Time Counter (RTC)            | 1                                             | 1                                             | 1                                             | 1                             |
| USART                              | 3                                             | 3                                             | 5                                             | 6                             |
| SPI                                | 2                                             | 2                                             | 2                                             | 2                             |
| TWI/I <sup>2</sup> C               | 1 <sup>(1)</sup>                              | 2 <sup>(1)</sup>                              | 2 <sup>(1)</sup>                              | 2 <sup>(1)</sup>              |
| 12-bit Differential ADC (channels) | 1 (10)                                        | 1 (14)                                        | 1 (18)                                        | 1 (22)                        |
| 10-bit DA(S)C (outputs)            | 1(1)                                          | 1(1)                                          | 1(1)                                          | 1(1)                          |
| Analog Comparator (AC)             | 3                                             | 3                                             | 3                                             | 3                             |
| Zero-Cross Detectors (ZCD)         | 1                                             | 1                                             | 2                                             | 3                             |

#### 22. TCB - 16-Bit Timer/Counter Type B

#### 22.1 Features

- 16-bit Counter Operation Modes:
  - Periodic interrupt
  - Time-out check
  - Input capture
    - On event
    - Frequency measurement
    - · Pulse-width measurement
    - Frequency and pulse-width measurement
    - 32-bit capture
  - Single-shot
  - 8-bit Pulse-Width Modulation (PWM)
- Noise Canceler on Event Input
- Synchronize Operation with TCAn

- Hardware can stop CPU from executing instructions and jump to a new place in memory and execute from there
  - New piece of code is called an Interrupt Service Routine (ISR)
  - Once ISR is complete, CPU jumps back to where it was previously and continues where it left off
- Perks
  - We can execute code on specific conditions
  - Especially useful for time sensitive operations

- Typical use cases
  - Communication
    - Example: Received data or ready for new data
  - Periodic interrupts
    - Example: Periodic measurements or actions
  - Safety critical hardware
    - Example: Brownout or Watchdog timeout
  - Measurements
    - Example: ADC conversion complete or ADC result above threshold
  - IO
    - Example: Change in IO logic
- AVR Event system
  - Some interrupts can be replaced by hardware in AVRs
  - Example: Start ADC on timer interrupt

- Pro
  - Perfect for time sensitive operations
  - Perfect for system timers
  - Perfect for rare occurrences
    - Do not need to check in main loop
  - Can prevent blocking code

- Con
  - Easy to abuse
  - Jumping from main.c at the wrong time can cause problems
    - Non-atomic instructions
    - Maybe a signal is left on too long?
  - ISR nesting
    - Timing issues
    - Race conditions
    - Deadlock
    - Stack depth

- Rules of thumb
  - Only use interrupts if it makes sense
    - Avoid using them to make things easy
  - Make ISRs short
    - Don't have many instructions in an ISR
    - Typical case: Read some register value, set a flag, clear interrupt, then back to main
  - Don't use interrupts for events that occur too frequently
    - SysTick is an exception
    - May become locked inside ISR
      - Round Robin
- If some interrupts are more time sensitive than others
  - Make them a higher priority!
  - Higher priority ISRs can interrupt other ISRs

- If some parts of your code should not be interrupted disable interrupts before doing these instructions and enable them after
- AVR
  - Global interrupt enable: sei()
  - Global interrupt disable: cli()
  - avr/interrupt.h must be included

# Interrupts in practice

#### Small ISR = OK

#### Software flag = GOOD

```
volatile uint8_t tcb0_flag = 0;
|int main(void)
    while (1)
        if (tcb0_flag)
            tcb0_flag = 0;
            // Do something
ISR(TCB0_INT_vect)
    // Clear interrupt flag
    TCB0.INTFLAGS = TCB_CAPT_bm;
    // Set software flag
    tcb0_flag = 1;
```

#### Long ISR and use of functions = BAD

```
ISR(TCB0_INT_vect)
{
    // Clear interrupt flag
    TCB0.INTFLAGS = TCB_CAPT_bm;
    check_adc();
    compute_pid_values();
    check_buttons();
}
```

- LED toggle task
  - No \_delay\_ms()
  - Instead we use software
- Same as Arduino function millis()
- Non blocking!
- Many tasks can use the same system timer
- Of course we need to configure TCB0 to have an interrupt every 1 ms

```
volatile uint32 t millis = 0;
int main(void)
    uint32 t led millis last = 0, led millis delay = 500;
    while (1)
        // Check if 500 ms has passed
        if ((millis - led millis last) >= led millis delay)
            // Update last millisecond value
            led millis last = millis;
            PORTC.OUTTGL = PIN6 bm;
ISR(TCB0 INT vect)
    // Clear interrupt flag
    TCB0.INTFLAGS = TCB CAPT bm;
    millis++;
```

- To do
  - Select clk\_tcb0 = F\_CPU/1
  - Select CNTMODE = INT
  - Enable CAPT interrupt
  - Set TOP value
    - TOP = CCMP

#### 22.3.3.1.1 Periodic Interrupt Mode

In the Periodic Interrupt mode, the counter counts to the capture value and restarts from BOTTOM. A CAPT interrupt and event is generated when the CNT is equal to TOP. If TOP is updated to a value lower than CNT, upon reaching MAX, an OVF interrupt and event is generated, and the counter restarts from BOTTOM.

Figure 22-3. Periodic Interrupt Mode



#### 22.4 Register Summary

| Offset | Name     | Bit Pos. | 7         | 6        | 5        | 4       | 3      | 2           | 1           | 0      |  |  |  |
|--------|----------|----------|-----------|----------|----------|---------|--------|-------------|-------------|--------|--|--|--|
| 0x00   | CTRLA    | 7:0      |           | RUNSTDBY | CASCADE  | SYNCUPD |        | CLKSEL[2:0] |             | ENABLE |  |  |  |
| 0x01   | CTRLB    | 7:0      |           | ASYNC    | CCMPINIT | CCMPEN  |        |             | CNTMODE[2:0 |        |  |  |  |
| 0x02   |          |          |           |          |          |         |        |             |             |        |  |  |  |
|        | Reserved |          |           |          |          |         |        |             |             |        |  |  |  |
| 0x03   |          |          |           |          |          |         |        |             |             |        |  |  |  |
| 0x04   | EVCTRL   | 7:0      |           | FILTER   |          | EDGE    |        |             |             | CAPTEI |  |  |  |
| 0x05   | INTCTRL  | 7:0      |           |          |          |         |        |             | OVF         | CAPT   |  |  |  |
| 0x06   | INTFLAGS | 7:0      |           |          |          |         |        |             | OVF         | CAPT   |  |  |  |
| 0x07   | STATUS   | 7:0      |           |          |          |         |        |             |             | RUN    |  |  |  |
| 0x08   | DBGCTRL  | 7:0      |           |          |          |         |        |             |             | DBGRUN |  |  |  |
| 0x09   | TEMP     | 7:0      |           |          |          | TEMI    | P[7:0] |             |             |        |  |  |  |
| 0x0A   | CNT      | 7:0      |           | CNT[7:0] |          |         |        |             |             |        |  |  |  |
| OXOA   | CIVI     | 15:8     | CNT[15:8] |          |          |         |        |             |             |        |  |  |  |
| 0x0C   | CCMP     | 7:0      |           |          |          | CCM     | P[7:0] |             |             |        |  |  |  |
| OXOC   | CCIVIF   | 15:8     |           |          |          | CCMF    | [15:8] |             |             |        |  |  |  |

- f = (clk\_cpu / DIV) / TOP
- TOP = clk\_cpu / (DIV \* f)
- f = 1000 [Hz] = 1/1000 [s]
- TOP = 4e6 / (1 \* 1000) = 4e3
  - This is within the bounds of 16 bits
  - $2^16-1 = 65535$
- Technically 1 clock cycle is used to restart the counter
  - Let's ignore that for now

#### 22.5.10 Capture/Compare

 Name:
 CCMP

 Offset:
 0x0C

 Reset:
 0x00

 Property:

The TCBn.CCMPL and TCBn.CCMPH register pair represents the 16-bit value TCBn.CCMP. The low byte [7:0] (suffix L) is accessible at the original offset. The high byte [15:8] (suffix H) can be accessed at offset  $+ 0 \times 01$ .

This register has different functions depending on the mode of operation:

- For Capture operation, these registers contain the captured value of the counter at the time the capture occurs
- In Periodic Interrupt, Time-Out Check and Single-Shot mode, this register acts as the TOP value
- In 8-bit PWM mode, TCBn.CCMPL and TCBn.CCMPH act as two independent registers: The period of the waveform is controlled by CCMPL, while CCMPH controls the duty cycle.

| Bit    | 15        | 14  | 13  | 12   | 11     | 10  | 9   | 8   |  |  |  |
|--------|-----------|-----|-----|------|--------|-----|-----|-----|--|--|--|
|        |           |     |     | CCMF | [15:8] |     |     |     |  |  |  |
| Access | R/W       | R/W | R/W | R/W  | R/W    | R/W | R/W | R/W |  |  |  |
| Reset  | 0         | 0   | 0   | 0    | 0      | 0   | 0   | 0   |  |  |  |
|        |           |     |     |      |        |     |     |     |  |  |  |
| Bit    | 7         | 6   | 5   | 4    | 3      | 2   | 1   | 0   |  |  |  |
| Г      | CCMP[7:0] |     |     |      |        |     |     |     |  |  |  |
| Access | R/W       | R/W | R/W | R/W  | R/W    | R/W | R/W | R/W |  |  |  |
| Reset  | 0         | 0   | 0   | 0    | 0      | 0   | 0   | 0   |  |  |  |

Bits 15:8 - CCMP[15:8] Capture/Compare Value High Byte

These bits hold the MSB of the 16-bit compare, capture and top value.

#### Bits 7:0 - CCMP[7:0] Capture/Compare Value Low Byte

These bits hold the LSB of the 16-bit compare, capture and top value.

Offset

0x00

0x01 0x02

#### 22.4 Register Summary

Name

CTRLA

CTRLB

Bit Pos.

7:0

7:0

6

RUNSTDBY

**ASYNC** 

CASCADE

CCMPINIT

SYNCUPD

CCMPEN

3

CLKSEL[2:0]

**ENABLE** 

CAPTEI

CAPT

RUN DBGRUN

CNTMODE[2:0]

OVF

OVF

|                                                                                                                                                                                           | <br>0x03<br>0x04 | Reserved     | 7:0     |           | FILTER   |         | EDGE     |         |
|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|------------------|--------------|---------|-----------|----------|---------|----------|---------|
| // Set PC6 as output                                                                                                                                                                      | 0x05             | INTCTRL      | 7:0     |           |          |         |          |         |
| •                                                                                                                                                                                         | 0x06             | INTFLAGS     | 7:0     |           |          |         |          |         |
| PORTC.DIRSET = PIN6_bm;                                                                                                                                                                   | 0x07             | STATUS       | 7:0     |           |          |         |          |         |
|                                                                                                                                                                                           | 0x08             | DBGCTRL      | 7:0     |           |          |         |          |         |
| // Set CNTMODE for TCB                                                                                                                                                                    | 0x09             | TEMP         | 7:0     |           |          |         | TEM      | P[7:0]  |
| TCBO.CTRLB = TCB CNTMODE INT gc;                                                                                                                                                          | 0x0A             | CNT          | 7:0     |           |          |         | CNT      | [7:0]   |
|                                                                                                                                                                                           | UXUA             | CIVI         | 15:8    |           | CNT[15:  |         |          |         |
| // Enable TCB0 overflow interrupt                                                                                                                                                         | 0x0C             | ССМР         | 7:0     |           |          |         | CCM      | P[7:0]  |
| •                                                                                                                                                                                         | UXUC             | CCIVIF       | 15:8    |           |          |         | CCMF     | P[15:8] |
| <pre>TCB0.INTCTRL = TCB_CAPT_bm;  // Calculate the TCB TOP Value TCB0.CCMP = (uint16_t)( ((float)F_CPU  // Enable TCB and set clock source to F_CPU TCB0.CTRLA = TCB_CLKSEL_DIV1_gc</pre> |                  | 000.0) - 0.5 | ); Alte | ernativel | y set TO | CBO.CCM | IP = 400 | 00      |
| ICB_ENABLE_DM;                                                                                                                                                                            |                  |              |         |           |          |         |          |         |

```
#define F CPU 4000000
#include <avr/io.h>
                                                                      uint32 t led millis last = 0, led millis delay = 500;
#include <avr/interrupt.h>
                                                                      while (1)
volatile uint32_t millis = 0;
                                                                          // Check if 500 ms has passed
jint main(void)
                                                                          if ((millis - led_millis_last) >= led_millis_delay)
    // Set PC6 as output
                                                                              // Update last millisecond value
    PORTC.DIRSET
                   = PIN6 bm;
                                                                              led_millis_last = millis;
    // Set CNTMODE for TCB
                                                                              PORTC.OUTTGL = PIN6 bm;
    TCB0.CTRLB
                    = TCB CNTMODE INT gc;
    // Enable TCB0 overflow interrupt
    TCB0.INTCTRL
                    = TCB_CAPT_bm;
                                                                      ISR(TCB0 INT vect)
    // Calculate the TCB TOP Value
    TCB0.CCMP
                    = (uint16_t)( ((float)F_CPU) / (1000.0) - 0.5 ); {
                                                                          // Clear interrupt flag
    // Enable TCB and set clock source to F_CPU DIV1
                                                                          TCB0.INTFLAGS = TCB_CAPT_bm;
    TCB0.CTRLA
                    = TCB CLKSEL DIV1 gc
                    TCB ENABLE bm;
                                                                          millis++;
    // Set Enable (Global) Interrupts
    sei();
```

- Final task for today
- Use TCA to control a servomotor
- TCA is bit more complicated than TCB
  - Fear not, it is actually not that hard to get going
  - Most of these registers can be ignored for now

#### 21.4 Register Summary - Single Mode

| Offset           | Name     | Bit Pos.    | 7        | 6           | 5      | 4      | 3                   | 2           | 1           | 0      |
|------------------|----------|-------------|----------|-------------|--------|--------|---------------------|-------------|-------------|--------|
| 0x00             | CTRLA    | 7:0         | RUNSTDBY |             |        |        |                     | CLKSEL[2:0] |             | ENABLE |
| 0x01             | CTRLB    | 7:0         |          | CMP2EN      | CMP1EN | CMP0EN | ALUPD               |             | WGMODE[2:0] |        |
| 0x02             | CTRLC    | 7:0         |          |             |        |        |                     | CMP2OV      | CMP1OV      | CMP0OV |
| 0x03             | CTRLD    | 7:0         |          |             |        |        |                     |             |             | SPLITM |
| 0x04             | CTRLECLR | 7:0         |          |             |        |        | CMI                 | D[1:0]      | LUPD        | DIR    |
| 0x05             | CTRLESET | 7:0         |          |             |        |        | CMI                 | D[1:0]      | LUPD        | DIR    |
| 0x06             | CTRLFCLR | 7:0         |          |             |        |        | CMP2BV              | CMP1BV      | CMP0BV      | PERBV  |
| 0x07             | CTRLFSET | 7:0         |          |             |        |        | CMP2BV              | CMP1BV      | CMP0BV      | PERBV  |
| 0x08             | Reserved |             |          |             |        |        |                     |             |             |        |
| 0x09             | EVCTRL   | 7:0         |          | EVACTB[2:0] |        | CNTBEI |                     | EVACTA[2:0] |             | CNTAEI |
| 0x0A             | INTCTRL  | 7:0         |          | CMP2        | CMP1   | CMP0   |                     |             |             | OVF    |
| 0x0B             | INTFLAGS | 7:0         |          | CMP2        | CMP1   | CMP0   |                     |             |             | OVF    |
| 0x0C<br><br>0x0D | Reserved |             |          |             |        |        |                     |             |             |        |
| 0x0E             | DBGCTRL  | 7:0         |          |             |        |        |                     |             |             | DBGRUN |
| 0x0F             | TEMP     | 7:0         |          |             |        | TEM    | P[7:0]              |             |             |        |
| 0x10<br><br>0x1F | Reserved |             |          |             |        |        |                     |             |             |        |
| 0x20             | CNT      | 7:0         |          |             |        |        | [7:0]               |             |             |        |
|                  |          | 15:8        |          |             |        | CNT[   | 15:8]               |             |             |        |
| 0x22<br><br>0x25 | Reserved |             |          |             |        |        |                     |             |             |        |
| 0x26             | PER      | 7:0         |          |             |        | PER    | [7:0]               |             |             |        |
| UX26             | PER      | 15:8        |          |             |        | PER[   | 15:8]               |             |             |        |
| 0x28             | CMP0     | 7:0         |          |             |        | CMP    | [7:0]               |             |             |        |
| UX28             | CMPU     | 15:8        |          |             |        | CMP    | [15:8]              |             |             |        |
| 0.24             | CMP1     | 7:0         |          |             |        | CMP    | [7:0]               |             |             |        |
| 0x2A             | CMP1     | 15:8        |          |             |        | CMP    | 15:8]               |             |             |        |
|                  |          | 7:0         |          |             |        | CMP    | [7:0]               |             |             |        |
| 0x2C             | CMP2     | 15:8        |          |             |        | CMP    | 15:8]               |             |             |        |
| 0x2E             |          |             |          |             |        |        |                     |             |             |        |
| <br>0x35         | Reserved |             |          |             |        |        |                     |             |             |        |
| 0.26             | DEDDLIE  | 7:0         |          |             |        | PERBL  | JF[7:0]             |             |             |        |
| 0x36             | PERBUF   | 15:8        |          |             |        | PERBU  | F[15:8]             |             |             |        |
| 020              | CMDODUS  | 7:0         |          |             |        |        | JF[7:0]             |             |             |        |
| 0x38             | CMP0BUF  | 15:8        |          |             |        | CMPBU  |                     |             |             |        |
|                  |          | 7:0         |          |             |        |        | JF[7:0]             |             |             |        |
|                  | CMP1BUF  |             |          |             |        |        |                     |             |             |        |
| 0x3A             |          | 15:8        |          |             |        | CMPBU  | JH115:81            |             |             |        |
| 0x3A<br>0x3C     | CMP2BUF  | 15:8<br>7:0 |          |             |        | CMPBU  | JF[15:8]<br>JF[7:0] |             |             |        |

- Controlling a servo motor
  - Pulse-Width controlled
    - On-time determines position
    - Dependent on servomotor
    - Typically 1000us to 2000us
      - For our servo this is 500us to 2500us
  - Pulse repetition
    - Usually 20 ms = 50 Hz
- Our servomotor has three pins
  - 5V
  - GND
  - SIG This is the PWM signal that controls the position

|                       |                       | Selection (           | Guide for Clu                                   | tch Servo                                       |                                                  |                                                  |
|-----------------------|-----------------------|-----------------------|-------------------------------------------------|-------------------------------------------------|--------------------------------------------------|--------------------------------------------------|
| Model                 | 6kg 180°              | 6kg 300°              | 9g 180°                                         | 9g 300°                                         | 2kg 180°                                         | 2kg 300°                                         |
| sku                   | SER0051               | SER0057               | SER0049                                         | SER0053                                         | SER0050                                          | SER0056                                          |
| Operating<br>Voltage  | 4.8-6VDC              | 4.8-6V DC             | 4.8-6V DC                                       | 4.8-6V DC                                       | 4.8-6V DC                                        | 4.8-6V DC                                        |
| Quiescent<br>Current  | ≤10mA at<br>6.0V      | ≤10mA at 6.0V         | ≤8mA at<br>6.0V                                 | ≤8mA at<br>6.0V                                 | ≤8mA at<br>6.0V                                  | ≤8mA at 6.0V                                     |
| No-load Curren        | ≤60mA at<br>6.0V      | ≤60mA at<br>6.0V      | ≤50mA at<br>4.8V<br>≤60mA at<br>6.0V            | ≤50mA at<br>4.8V<br>≤60mA at<br>6.0V            | ≤110mA at<br>4.8V<br>≤120mA at<br>6.0V           | ≤110mA at<br>4.8V<br>≤120mA at<br>6.0V           |
| Stall Current         | ≤1.65A at<br>6.0V     | ≤1.65A at<br>6.0V     | ≤550mA at<br>4.8V<br>≤650mA at<br>6.0V          | ≤550mA at<br>4.8V<br>≤650mA at<br>6.0V          | ≤700mA at<br>4.8V<br>≤800mA at<br>6.0V           | ≤700mA at<br>4.8V<br>≤800mA at<br>6.0V           |
| Rated Torque          | ≥4.4kg.com<br>at 6.0V | ≥4.4kg.com<br>at 6.0V | ≥0.32kgf·cm<br>at 4.8V<br>≤0.4kgf·cm<br>at 6.0V | ≥0.32kgf·cm<br>at 4.8V<br>≤0.4kgf·cm<br>at 6.0V | ≥0.45kgf·cm<br>at 4.8V<br>≥0.55kgf·cm<br>at 6.0V | ≥0.45kgf·cm<br>at 4.8V<br>≥0.55kgf·cm<br>at 6.0V |
| Stall Torque          | ≥6kg·cm at<br>6.0V    | ≥6kg·cm at<br>6.0V    | ≥1.0kgf·cm<br>at 4.8V<br>≤1.2kgf·cm<br>at 6.0V  | ≥1.0kgf·cm<br>at 4.8V<br>≤1.2kgf·cm<br>at 6.0V  | ≥1.6kgf·cm<br>at 4.8V<br>≥2.0kgf·cm<br>at 6.0V   | ≥1.6kgf·cm at<br>4.8V<br>≥2.0kgf·cm at<br>6.0V   |
| Operating<br>Angle    | 180°±10°              | 300°±10°              | 180°±10°                                        | 300°±10°                                        | 180°±10°                                         | 300°±10°                                         |
| Pulse Width<br>Range  | 500~2500μs            | 500~2500μs            | 500~2500μs                                      | 500~2500μs                                      | 500~2500μs                                       | 500~2500μs                                       |
| Communication<br>Mode | PWM                   | PWM                   | PWM                                             | PWM                                             | PWM                                              | PWM                                              |

- We know
  - Period = 20 ms = 50 Hz
  - On time = 500 to 2500 us
- To get the best resolution we need to have a large TOP value
  - TOP = f\_clk\_tca / f\_pwm
  - For f\_clk\_tca = 4 MHz we get: 4e6 / 50 = 80000 This is larger than 2^16 !!!
  - For f\_clk\_tca = 2 MHz we get 40000 < 2^16
- Prescaler value must be 2 when F\_CPU = 4 MHz
- TOP must be 40000

#### Now for ON time

- 500 us = 2000 Hz
  - CMP = 2 MHz / 2000 Hz = 1000
- 2500 us = 400 Hz
  - CMP = 2 MHz / 400 Hz = 5000
- General formula
  - duty\_cycle = 0 to 100
  - CMP = duty\_cycle\*((5000-1000)/100) + 1000
  - Simplify: CMP = duty\_cycle\*40 + 1000
  - Sanity check

- Create a function that solves CMP value
  - TCA can use double buffering
  - By using double buffering the PWM signal is always valid

```
void servo_set(uint8_t duty_cycle)
{
    // Optional guard
    if (duty_cycle > 100)
    {
        duty_cycle = 100;
    }

    // Calculate compare value
    uint16_t cmp = duty_cycle * 40 + 1000;

    // Update CMPxBUF register. CMPx register us updated by buffer on overflow
    // This ensures that the pulse on time is always valid
    TCA0.SINGLE.CMP0BUF = cmp;
}
```

TCA is a unique peripheral as its registers are different depending on if it is in SPLIT mode, or SINGLE mode! We must therefore use TCAx.SINGLE.PERIPHERAL

- Setup
  - Calculate and set TOP value (PER)
    - Already calculated!
  - Pin override enable (CTRLB)
    - Make sure TCA can override OUT value for a physical pin
  - Set TCA to PWM mode (CTRLB)
    - WGMODE bits
  - Select prescaler value (CTRLA)
    - CLKSEL bits
  - Enable TCA instance (CTRLA)
    - ENABLE bit

#### 21.4 Register Summary - Single Mode

| 0x01         CTRLB         7:0         CMP2EN         CMP1EN         CMP0EN         ALUPD         WGMODE[2:0]           0x02         CTRLC         7:0         CMP2OV         CMP1OV         CMP0OV           0x03         CTRLD         7:0         CMD[1:0]         LUPD         DIR           0x04         CTRLESET         7:0         CMD[1:0]         LUPD         DIR           0x05         CTRLESET         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x06         CTRLFSET         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x08         Reserved         0x08         EVCTRL         7:0         EVACTB[2:0]         CNTBEI         EVACTA[2:0]         CNTAEI           0x0A         INTCTRL         7:0         CMP2         CMP1         CMP0         OVF           0x0C          Reserved         OXD         CMP1         CMP0         OVF                                                                                                                                                                                                                                                                                          | Offset | Name     | Bit Pos. | 7        | 6           | 5      | 4      | 3      | 2           | 1           | 0      |
|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--------|----------|----------|----------|-------------|--------|--------|--------|-------------|-------------|--------|
| Ox02                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                               | 0x00   | CTRLA    | 7:0      | RUNSTDBY |             |        |        |        | CLKSEL[2:0] |             | ENABLE |
| 0x03         CTRLD         7:0         SPLITM           0x04         CTRLECLR         7:0         CMD[1:0]         LUPD         DIR           0x05         CTRLESET         7:0         CMD[1:0]         LUPD         DIR           0x06         CTRLFCLR         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x07         CTRLFSET         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x08         Reserved         CMP2         CMP1         CMP2BV         CMP1BV         CMP0BV         PERBV           0x09         EVCTRL         7:0         EVACTB[2:0]         CNTBEI         EVACTA[2:0]         CNTAEI           0x0A         INTCTRL         7:0         CMP2         CMP1         CMP0         OVF           0x0C          Reserved         OVF         OVF         OVF         OVF           0x0D         DBGCTRL         7:0         TEMP[7:0]         TEMP[7:0]         OVE         OVE           0x1D         Reserved         CNT[15:8]                             | 0x01   | CTRLB    | 7:0      |          | CMP2EN      | CMP1EN | CMP0EN | ALUPD  |             | WGMODE[2:0] |        |
| 0x04         CTRLECLR         7:0         CMD[1:0]         LUPD         DIR           0x05         CTRLESET         7:0         CMD[1:0]         LUPD         DIR           0x06         CTRLFCLR         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x07         CTRLFSET         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x08         Reserved         CMP2         CMP1         CMP0         CNTAEI         CNTAEI           0x0A         INTCTRL         7:0         CMP2         CMP1         CMP0         OVF           0x0B         INTFLAGS         7:0         CMP2         CMP1         CMP0         OVF           0x0C          Reserved         OXOB         TEMP(7:0]         DBGRUN           0x0F         TEMP         7:0         TEMP(7:0]         CNT[7:0]           0x1G          CNT[7:0]         CNT[15:8]           0x2D          Reserved         CNT[15:8]           0x2D          Reserved          CNT[7:0]           0x2D          CNT          CNT[7:0]           0x2D <t< td=""><td>0x02</td><td>CTRLC</td><td>7:0</td><td></td><td></td><td></td><td></td><td></td><td>CMP2OV</td><td>CMP1OV</td><td>CMP0OV</td></t<> | 0x02   | CTRLC    | 7:0      |          |             |        |        |        | CMP2OV      | CMP1OV      | CMP0OV |
| 0x05         CTRLESET         7:0         CMD[1:0]         LUPD         DIR           0x06         CTRLFCLR         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x07         CTRLFSET         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x08         Reserved         CMP2BV         CMP1BV         CMP0BV         PERBV           0x09         EVCTRL         7:0         EVACTB[2:0]         CNTAEI         EVACTA[2:0]         CNTAEI           0x0A         INTCTRL         7:0         CMP2         CMP1         CMP0         OVF           0x0C          Reserved         OVF         OVF         OVF           0x0E         DBGCTRL         7:0         CMP2         CMP1         CMP0         DBGRUN           0x1D          Reserved         OX1D         CNT[7:0]         CNT[7:0]         CNT[7:0]         CNT[7:0]         CNT[15:8]         CNT[15:8]         CNT[15:8]         CNT[7:0]         CNT[        | 0x03   | CTRLD    | 7:0      |          |             |        |        |        |             |             | SPLITM |
| 0x06         CTRLFCLR         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x07         CTRLFSET         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x08         Reserved         CMP1BV         CMP1BV         CMP0BV         PERBV           0x09         EVCTRL         7:0         EVACTB[2:0]         CNTBEI         EVACTA[2:0]         CNTAEI           0x0A         INTCTRL         7:0         CMP2         CMP1         CMP0         OVF           0x0B         INTFLAGS         7:0         CMP2         CMP1         CMP0         OVF           0x0B         DBGCTRL         7:0         TEMP[7:0]         DBGRUN           0x1D         Reserved         TEMP[7:0]         CNT[7:0]           0x1D         TEMP         7:0         CNT[7:0]           0x2D         CNT         TS:8         CNT[15:8]           0x2D         Reserved         TS:8         CNT[15:8]                                                                                                                                                                                                                                                                       | 0x04   | CTRLECLR | 7:0      |          |             |        |        | CME    | [1:0]       | LUPD        | DIR    |
| 0x07         CTRLFSET         7:0         CMP2BV         CMP1BV         CMP0BV         PERBV           0x08         Reserved         0x09         EVCTRL         7:0         EVACTB[2:0]         CNTBEI         EVACTA[2:0]         CNTAEI           0x0A         INTERING         7:0         CMP2         CMP1         CMP0         OVF           0x0B         DBGCTRL         7:0         CMP2         CMP1         CMP0         DBGRUN           0x0F         TEMP         7:0         CNT[7:0]           0x1D          Reserved           0x1F         Ox20         CNT         7:0         CNT[15:8]           0x22          Reserved           0x25          PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       | 0x05   | CTRLESET | 7:0      |          |             |        |        | CME    | [1:0]       | LUPD        | DIR    |
| 0x08         Reserved           0x09         EVCTRL         7:0         EVACTB[2:0]         CNTBEI         EVACTA[2:0]         CNTAEI           0x0A         INTCRL         7:0         CMP2         CMP1         CMP0         OVF           0x0B         INTELAGS         7:0         CMP2         CMP1         CMP0         OVF           0x0C          Reserved         Reserved         DBGRUN           0x0F         TEMP         7:0         TEMP[7:0]           0x10          Reserved         CNT[7:0]           0x1F         CNT[15:8]         CNT[15:8]           0x22          Reserved           0x25         PER         7:0         PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         | 0x06   | CTRLFCLR | 7:0      |          |             |        |        | CMP2BV | CMP1BV      | CMP0BV      | PERBV  |
| 0x09         EVCTRL         7:0         EVACTB[2:0]         CNTBEI         EVACTA[2:0]         CNTAEI           0x0A         INTCTRL         7:0         CMP2         CMP1         CMP0         OVF           0x0B         INTFLAGS         7:0         CMP2         CMP1         CMP0         OVF           0x0C          Reserved         Reserved         DBGCTRL         7:0         DBGRUN           0x10          Reserved         Reserved         CNT[7:0]         CNT[7:0]           0x20         CNT         T5:8         CNT[15:8]         CNT[15:8]           0x22          Reserved         PER         7:0         PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          | 0x07   | CTRLFSET | 7:0      |          |             |        |        | CMP2BV | CMP1BV      | CMP0BV      | PERBV  |
| 0x0A         INTCTRL         7:0         CMP2         CMP1         CMP0         OVF           0x0B         INTFLAGS         7:0         CMP2         CMP1         CMP0         OVF           0x0C          Reserved         Reserved         OVF         OVF         OVF           0x0E         DBGCTRL         7:0         TEMP[7:0]         TEMP[7:0]         OVF                                    | 0x08   | Reserved |          |          |             |        |        |        |             |             |        |
| 0x0B         INTFLAGS         7:0         CMP2         CMP1         CMP0         OVF           0x0C          Reserved         0x0D         DBGCTRL         7:0         DBGRUN           0x0F         TEMP         7:0         TEMP[7:0]         TEMP[7:0]           0x10          Reserved         CNT[7:0]         CNT[7:0]           0x20         CNT         15:8         CNT[15:8]           0x22          Reserved         PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           | 0x09   | EVCTRL   | 7:0      |          | EVACTB[2:0] |        | CNTBEI |        | EVACTA[2:0] |             | CNTAEI |
| 0x0C          Reserved           0x0D         0x0E         DBGCTRL         7:0         DBGRUN           0x0F         TEMP         7:0         TEMP[7:0]           0x10          Reserved         Reserved         CNT[7:0]           0x20         CNT         7:0         CNT[15:8]         CNT[15:8]           0x22          Reserved         Reserved         PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           | 0x0A   | INTCTRL  | 7:0      |          | CMP2        | CMP1   | CMP0   |        |             |             | OVF    |
| Reserved                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           | 0x0B   | INTFLAGS | 7:0      |          | CMP2        | CMP1   | CMP0   |        |             |             | OVF    |
| 0x0F         TEMP         7:0         TEMP[7:0]           0x10          Reserved         CNT[7:0]           0x1F         7:0         CNT[7:0]           0x20         CNT         15:8         CNT[15:8]           0x22          Reserved         Reserved         PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         |        | Reserved |          |          |             |        |        |        |             |             |        |
| 0x10      Reserved       0x1F     7:0     CNT[7:0]       0x20     15:8     CNT[15:8]       0x22      Reserved       0x25     7:0     PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                      | 0x0E   | DBGCTRL  | 7:0      |          |             |        |        |        |             |             | DBGRUN |
| Reserved 0x1F  0x20                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                | 0x0F   | TEMP     | 7:0      |          |             |        | TEM    | P[7:0] |             |             |        |
| 0x20 CNT 15:8 CNT[15:8]  0x22 Reserved 0x25 PER 7:0 PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |        | Reserved |          |          |             |        |        |        |             |             |        |
| 15:8 CNT[15:8]  0x22 Reserved 0x25  0x26 PER 7:0 PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                          | 0×20   | CNT      | 7:0      |          |             |        | CNT    | [7:0]  |             |             |        |
| Reserved 0x25 7:0 PER[7:0]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                         | 0.00   | CIVI     | 15:8     |          |             |        | CNT[   | [15:8] |             |             |        |
| 0x26 PER                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                           |        | Reserved |          |          |             |        |        |        |             |             |        |
| 15:8 PER[15:8]                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                     | 0x26   | PER      | 7:0      |          |             |        | PER    | [7:0]  |             |             |        |
|                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    | UXZU   | PER      |          |          |             |        | _      | -      |             |             |        |

- Find which pins are physically connected to a TCA instance
- Development board pinout
  - PD0 PD2 are optional connections (Multiplexed)
- Default connections
  - PA0 PA2
  - We will use PA0
- WOn
  - WO0, WO1, WO2 available in all modes
  - WO3, WO4, WO5 available ONLY in SPLIT mode



| VQFN64/<br>TQFP64 | VQFN48/<br>TQFP48 | VQFN32/<br>TQFP32 | SOIC28/<br>SOIC28/ | Pin name (1,2) | Spedal | ADC0 | PTC     | ACı                     | DACO | ZCDn                    | USARTn                | SPIn                           | TWIn(4)                 | TCA0               | ξĪ  | TCBn                | TCD                  | EVSYS      | CCL-LUTIN            |
|-------------------|-------------------|-------------------|--------------------|----------------|--------|------|---------|-------------------------|------|-------------------------|-----------------------|--------------------------------|-------------------------|--------------------|-----|---------------------|----------------------|------------|----------------------|
| 62                | 44                | 30                | 22                 | PA0            | EXTCLK |      | XO/Y0   |                         |      |                         | 0,TxD                 |                                |                         | WO0                |     |                     |                      |            | 0,IN0                |
| 63                | 45                | 31                | 23                 | PA1            |        |      | X1/Y1   |                         |      |                         | 0,RxD                 |                                |                         | WO1                |     |                     |                      |            | 0,IN1                |
| 64                | 46                | 32                | 24                 | PA2            | TWI    |      | X2/Y2   |                         |      |                         | 0,XCK                 |                                | 0,SDA(H)                | WO2                |     | 0,WO                |                      | EVOUTA     | 0,IN2                |
| 1                 | 47                | 1                 | 25                 | PA3            | TWI    |      | X3/Y3   |                         |      |                         | 0,XDIR                |                                | 0,SCL(H)                | WO3                |     | 1,WO                |                      |            | 0,OUT                |
| 2                 | 48                | 2                 | 26                 | PA4            |        |      | X4/Y4   |                         |      |                         | 0,TxD <sup>(3)</sup>  | 0,MOSI                         |                         | WO4                |     |                     | 0,WOA                |            |                      |
| 3                 | 1                 | 3                 | 27                 | PA5            |        |      | X5/Y5   |                         |      |                         | 0,RxD(3)              | 0,MISO                         |                         | WO5                |     |                     | 0,WOB                |            |                      |
| 4                 | 2                 | 4                 | 28                 | PA6            |        |      | X6/Y6   |                         |      |                         | 0,XCK <sup>(3)</sup>  | 0,SCK                          |                         |                    |     |                     | 0,WOC                |            | 0,OUT <sup>(3)</sup> |
| 5                 | 3                 | 5                 | 1                  | PA7            | CLKOUT |      | X7/Y7   | 0,OUT<br>1,OUT<br>2,OUT |      | 0,OUT<br>1,OUT<br>2,OUT | 0,XDIR <sup>(3)</sup> | 0,55                           |                         |                    |     |                     | 0,WOD                | EVOUTA (3) |                      |
| 6                 |                   |                   |                    | VDD            |        |      |         |                         |      |                         |                       |                                |                         |                    |     |                     |                      |            |                      |
| 7                 |                   |                   |                    | GND            |        |      |         |                         |      |                         |                       |                                |                         |                    |     |                     |                      |            |                      |
| 8                 | 4                 |                   |                    | PB0            |        |      | X8/Y8   |                         |      |                         | 3,TxD                 |                                |                         | MO0(3)             | WO0 |                     |                      |            | 4,IN0                |
| 9                 | 5                 |                   |                    | PB1            |        |      | X9/Y9   |                         |      |                         | 3,RxD                 |                                |                         | WO1 <sup>(3)</sup> | WO1 |                     |                      |            | 4,IN1                |
| 10                | 6                 |                   |                    | PB2            |        |      | X10/Y10 |                         |      |                         | 3,XCK                 |                                | 1,SDA(H) <sup>(3)</sup> | WO2 <sup>(3)</sup> | WO2 |                     |                      | EVOUTB     | 4,IN2                |
| - 11              | 7                 |                   |                    | PB3            |        |      | X11/Y11 |                         |      |                         | 3,XDIR                |                                | 1,SCL(H) <sup>(3)</sup> | WO3 <sup>(3)</sup> | WO3 |                     |                      |            | 4,OUT                |
| 12                | 8                 |                   |                    | PB4            |        |      | X12/Y12 |                         |      |                         | 3,TxD <sup>(3)</sup>  | 1,MOSI <sup>(3)</sup>          |                         | WO4 <sup>(3)</sup> | WO4 | 2,WO <sup>(3)</sup> | 0,WOA <sup>(3)</sup> |            |                      |
| 13                | 9                 |                   |                    | PB5            |        |      | X13/Y13 |                         |      |                         | 3,RxD(3)              | 1,MISO(3)                      |                         | WO5(3)             | WO5 | 3,WO                | 0,WOB(3)             |            |                      |
| 14                |                   |                   |                    | PB6            |        |      | X14/Y14 |                         |      |                         | 3,XCK <sup>(3)</sup>  | 1,SCK <sup>(3)</sup>           | 1,SDA(C) <sup>(3)</sup> |                    |     |                     | 0,WOC <sup>(3)</sup> |            | 4,OUT <sup>(3)</sup> |
| 15                |                   |                   |                    | PB7            |        |      | X15/Y15 |                         |      |                         | 3,XDIR <sup>(3)</sup> | 1, <del>SS<sup>(3)</sup></del> | 1,SCL(C) <sup>(3)</sup> |                    |     |                     | 0,WOD(3)             | EVOUTB (3) |                      |
| 16                | 10                | 6                 | 2                  | PC0            |        |      |         |                         |      |                         | 1,TxD                 | 1,MOSI                         |                         | WO0(3)             |     | 2,WO                |                      |            | 1,IN0                |
| 17                | 11                | 7                 | 2                  | DC1            |        |      |         |                         |      |                         | 1 DvD                 | 1 MICO                         |                         | WO1(3)             |     | 2 MO(3)             |                      |            | 1 1811               |

- TCA0 is used
  - We use TCA0 in SINGLE mode
    - Only WO0, WO1, WO2 available
  - We can connect three servomotors because WO0, WO1, WO2 are independent!
  - Let us connect to WOO

Figure 4-1. AVR128DA48 Curiosity Nano Pinout



- Connect servo
  - VBUS to Supply pin (RED)
    - VBUS is always 5.0 V
  - GND to GND (BROWN)
  - Control signal to PAO (YELLOW)
- GPIO setup
  - Make PAO an OUTPUT
    - Set DIR register
  - For TCA0 to override PA0 we must set CMP0EN in TCA0.CTRLB

Figure 4-1. AVR128DA48 Curiosity Nano Pinout



#### 21.5.2 Control B - Normal Mode

 Name:
 CTRLB

 Offset:
 0x01

 Reset:
 0x00

 Property:



#### Bits 4, 5, 6 - CMPEN Compare n Enable

In the FRQ and PWM Waveform Generation modes, the Compare n Enable (CMPnEN) bits will make the waveform output available on the pin corresponding to WOn.

| Value | Description                                                                 |
|-------|-----------------------------------------------------------------------------|
| 0     | Waveform output WOn will not be available on the corresponding pin          |
| 1     | Waveform output WOn will override the output value of the corresponding pin |

- To Do
  - Set PA0 as output
  - PER
    - Set period
  - CTRLB
    - Enable pin override on PAO
    - Set waveform mode to PWM single-slope
      - Only up-count
  - CTRLA
    - Set clock to F\_CPU / 2
    - Enable TCA0

#### Bits 2:0 - WGMODE[2:0] Waveform Generation Mode

This bit field selects the Waveform Generation mode and controls the counting sequence of the counter, TOP value, UPDATE condition, interrupt condition, and the type of waveform generated. No waveform generation is performed in the Normal mode of operation. The waveform generator output will only be directed to the port pins if setting the corresponding CMPnEN bit for all other modes. The port pin direction must be set as output.

Table 21-7. Timer Waveform Generation Mode

| Value | Group Configuration | Mode of Operation | TOP  | UPDATE | OVF            |
|-------|---------------------|-------------------|------|--------|----------------|
| 0x0   | NORMAL              | Normal            | PER  | TOP(1) | TOP(1)         |
| 0x1   | FRQ                 | Frequency         | CMP0 | TOP(1) | TOP(1)         |
| 0x2   | -                   | Reserved          | -    | -      | -              |
| 0x3   | SINGLESLOPE         | Single-slope PWM  | PER  | воттом | BOTTOM         |
| 0x4   | -                   | Reserved          | -    | -      | -              |
| 0x5   | DSTOP               | Dual-slope PWM    | PER  | воттом | TOP            |
| 0x6   | DSBOTH              | Dual-slope PWM    | PER  | воттом | TOP and BOTTOM |
| 0x7   | DSBOTTOM            | Dual-slope PWM    | PER  | воттом | BOTTOM         |

```
void servo setup()
    #define TCA CLKDIV 2
    #define TCA FREQ 50.0
   // Calculate TOP value for TCA
    TCA0.SINGLE.PER
                       = (uint16\ t)(((float)F\ CPU)/((float)TCA\ CLKDIV))/TCA\ FREQ - 0.5);
                       = TCA SINGLE CMP0EN bm
    TCA0.SINGLE.CTRLB
                                                           // Enable pin override on CMP0 - WO0 - PD0
                        TCA SINGLE WGMODE SINGLESLOPE gc; // TCA0 in PWM mode (Single slope)
                       = TCA_SINGLE_CLKSEL_DIV2_gc
    TCA0.SINGLE.CTRLA
                                                           // Set F TCA = F CPU/2
                        TCA_SINGLE_ENABLE_bm;
int main(void)
    PORTA.DIRSET
                   = PIN0 bm;
    servo_setup();
```

- Test Servo motor!
  - Let the motor operate in peace
    - Otherwise the fuse on the development board may pop

- Optional tasks:
  - Connect enable servo motor control on another GPIO
  - Make the servo change position on button press

```
int main(void)
    PORTA.DIRSET
                    = PIN0 bm;
    servo_setup();
    while (1)
       servo_set(0); // Servo 0%
       _delay_ms(500);
       servo_set(50); // Servo 50%
       _delay_ms(500);
       servo_set(100); // Servo 100%
       _delay_ms(500);
```

# Next time

- Learn how to use a basic sensor
- Learn how to use SPI to connect to other devices
- Focus on hands-on experience from next session

Questions?